-
Notifications
You must be signed in to change notification settings - Fork 236
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Injectivity of inductive types revisited #3253
Conversation
…high; dd support for --ext 'compat:injectivity' for assisting with breakages
then List.map2 (fun v a -> mkEq(mkFreeV v, a)) vars indices | ||
else [] in | ||
else ( | ||
//only injectivity on indices |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note, we always get injectivity on indices
I have merged PRs into zeta, steel, pulse, and everparse for compatibility with this PR. |
to a universe <= the universe of the constructed type. | ||
See BugBoxInjectivity.fst *) | ||
u_leq_u_k u | ||
| Tm_name _ -> (* this is a value of another type parameter in scope *) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This name is another (opened) type parameter? If that type parameter itself is a type in a higher universe, that's ok? As in, returning true here is ok in that case?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry, wrong branch.
FYI, I tested this PR on my projects, and the following sample of code do not typecheck anymore (regardless the fuel/ifuel/z3rlimit): #set-options "--fuel 0 --ifuel 0"
noeq type map_types = {
key: eqtype;
value: Type0;
}
noeq type map_elem (mt:map_types) = {
key: mt.key;
value: mt.value;
}
noeq type map (mt:map_types) = {
key_values: list (map_elem mt)
}
#push-options "--fuel 1 --ifuel 1"
val find_value_aux: #mt:map_types -> key:mt.key -> l:list (map_elem mt) -> Pure (option mt.value)
(requires True)
(ensures fun res ->
match res with
| None -> True
| Some value -> List.Tot.memP ({key; value;}) l
)
let rec find_value_aux #mt key l =
match l with
| [] -> None
| h::t ->
if h.key = key then
Some h.value
else
match find_value_aux key t with
| Some res -> Some res
| None -> None
#pop-options |
Thanks for the example @TWal This is indeed a problem, one that could eventually be remedied by what I propose as a long-term fix:
In the style you have, the parameter Here's another way to write it, in a curried style, that does work: #set-options "--fuel 0 --ifuel 0"
noeq type map_types = {
key: eqtype;
value: Type0;
}
noeq type map_elem' (k:eqtype) (v:Type0) = {
key: k;
value: v;
}
let map_elem (mt:map_types) = map_elem' mt.key mt.value
noeq type map' (k:eqtype) (v:Type0) = {
key_values: list (map_elem' k v)
}
let map (mt:map_types) = map' mt.key mt.value
#push-options "--fuel 1 --ifuel 1"
val find_value_aux: #mt:map_types -> key:mt.key -> l:list (map_elem mt) -> Pure (option mt.value)
(requires True)
(ensures fun res ->
match res with
| None -> True
| Some value -> List.Tot.memP ({key; value;}) l
)
let rec find_value_aux #mt key l =
match l with
| [] -> None
| h::t ->
if h.key = key then
Some h.value
else
match find_value_aux key t with
| Some res -> Some res
| None -> None Would this style work for your code? |
Thank you for the workaround! I guess that would work, although a bit inconvenient because the record I do have a few follow-up questions:
On a higher-level, I don't understand what is the purpose of that pull-request. Is it because there are still unsoundnesses in the injectivity despite the previous fixes? |
…ity of _data_ constructor on the type parameter
…axiom to prove that the parameter instantiations are irrelevant and remove the projector axiom
(fun n x -> | ||
let field_projectible = | ||
n >= n_tps || //either this field is not a type parameter | ||
is_injective_on_tparams //or we are allowed to be injective on parameters |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Type parameter arguments are only projectible when this is_injective_on_tparams is true
constr_fields=fields; | ||
constr_sort=Term_sort; | ||
constr_id=Some (varops.next_id()); | ||
constr_base=not is_injective_on_tparams |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Generate the base constructor for irrelevance of type param arguments, if not injective on type params
let elim_eqns_and_guards = | ||
List.fold_left2 | ||
(fun elim_eqns_and_guards data_arg_param arg_param -> | ||
Term.subst elim_eqns_and_guards data_arg_param arg_param) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This substitution is important: State the typing inversion axiom in terms of the parameters of the type constructor, not the type-parameter arguments of the data constructor---since the latter may be semantically irrelevant.
We need to also fix the implementation of type idx : Type u#2 = | A1 | A2
noeq
type test2 (i:idx) : Type u#1 =
| Mk2 : test2 i
let bad
(x0 : test2 A1) (x1 : test2 A2) : Lemma False =
assert (test2 A1 == test2 A2);
assert_norm (~(coerce_eq () (Mk2 #A1) == Mk2 #A2));
()
let falso () : Lemma False = bad Mk2 Mk2 There may be similar rules in tactics, I'll check. Also I was suprised to see the equality between |
Thanks @mtzguido! That's a great catch on Syntax.Util.
Indeed, yes! This is also a surprise and has to do with the encoding of pretyping axiom. I'm looking into it. |
For the equal_data thing, maybe the The tactics module seems fine. |
Thanks @mtzguido for your comment above on equality of data constructors and all the help over the past few days ... My latest set of commits have two additional fixes:
Aside from some rlimit and solve restart tweaks, I did not need further changes in the the everest repos |
…sts of data constructors
In Issue #349, we see that injective type formers are inconsistent.
In PR #1649, we have a fix to remove injectivity for inductive type definitions, but as remarked there, the fix was unsatisfactory since it only eliminates injectivity on type constructors with type-function parameters whose domain is too large. There are two problems with this:
The following equivalent variant would---we were not normalizing the type before the check, which is clearly wrong.
So, this PR first normalizes the type of the parameter.
But, of course, an injective function from Type u#1 -> Type u#0 is nonsense, and with Guido's FStar.Cardinality.Universes you can actually prove that and get False out of it. See BugBoxInjectivity.fst
So, now we also check that the codomain of the type parameter is in a universe less than or equal to the universe of the constructed type.
#push-options "--ext 'compat:injectivity'
to retain the behavior prior to the fix, to assist with porting code.The impact of this fix is as follows:
See FStar.Constructive.fst: We now need to explicitly destruct constructive equality proofs rather than letting the SMT solver do it, since now the axiom encoding the inversion principle is weaker.
The same kind of explicit destruction of proof witnesses also shows up in FStar.WellFounded, FStar.WellFounded.Util, FStar.WellFoundedRelation and FStar.ModifiesGen
In Steel & Pulse, we add meaningless parameters to some inductive types as a type-inference hints, e.g., we have
although that
t
is actually unused.I fixed the Pulse library Pulse.Lib.HigherArray that used this pattern and instead write it like
retaining the type parameter for better unification in
ptr t
, but in a shallow, transparent way. This required exposing core_ref and core_ghost_ref etc. in PulseCore, but that was relatively easy.The same fixes could be applied to Steel.ST.HigherArray, but I did not want to change SteelCore: so, I left this module as is, using
#push-options "--ext 'compat:injectivity'` around the relevant type definitions. Similarly, in Steel.Semantics.Hoare.MST and Steel.ST.EphemeralHashTbl, I used this compat option.
We have also been discussing a better long-term fix, which is to simply remove the type parameters altogether from the SMT encoding of a data constructor. That will simplify a lot of things, but is a significant change and will likely pose some challenges in generating the correct patterns for quantifiers, since the parameters will not be present. That change may also open the door to an even more long-standing issue, i.e., #65, subtyping for the parameters of an inductive.